home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Info-Mac 1992 August
/
info-mac-1992.iso
/
Applications (app)
/
STvi
/
stevie 3.10
/
main.c
< prev
next >
Wrap
C/C++ Source or Header
|
1991-01-03
|
7KB
|
340 lines
/* $Header: /nw/tony/src/stevie/src/RCS/main.c,v 1.12 89/08/02 19:53:27 tony Exp $
*
* The main routine and routines to deal with the input buffer.
*/
#include "stevie.h"
int Rows; /* Number of Rows and Columns */
int Columns; /* in the current window. */
char *Realscreen = NULL; /* What's currently on the screen, a single */
/* array of size Rows*Columns. */
char *Nextscreen = NULL; /* What's to be put on the screen. */
char *Filename = NULL; /* Current file name */
LPTR *Filemem; /* Pointer to the first line of the file */
LPTR *Filetop; /* Line 'above' the start of the file */
LPTR *Fileend; /* Pointer to the end of the file in Filemem. */
/* (It points to the byte AFTER the last byte.) */
LPTR *Topchar; /* Pointer to the byte in Filemem which is */
/* in the upper left corner of the screen. */
LPTR *Botchar; /* Pointer to the byte in Filemem which is */
/* just off the bottom of the screen. */
LPTR *Curschar; /* Pointer to byte in Filemem at which the */
/* cursor is currently placed. */
int Cursrow, Curscol; /* Current position of cursor */
int Cursvcol; /* Current virtual column, the column number of */
/* the file's actual line, as opposed to the */
/* column number we're at on the screen. This */
/* makes a difference on lines that span more */
/* than one screen line. */
int Curswant = 0; /* The column we'd like to be at. This is used */
/* try to stay in the same column through up/down */
/* cursor motions. */
bool_t set_want_col; /* If set, then update Curswant the next time */
/* through cursupdate() to the current virtual */
/* column. */
int State = NORMAL; /* This is the current state of the command */
/* interpreter. */
int Prenum = 0; /* The (optional) number before a command. */
LPTR *Insstart; /* This is where the latest insert/append */
/* mode started. */
bool_t Changed = 0; /* Set to 1 if something in the file has been */
/* changed and not written out. */
char Redobuff[1024]; /* Each command should stuff characters into this */
/* buffer that will re-execute itself. */
char Insbuff[1024]; /* Each insertion gets stuffed into this buffer. */
int Ninsert = 0; /* Number of characters in the current insertion. */
char *Insptr = NULL;
bool_t got_int=FALSE; /* set to TRUE when an interrupt occurs (if possible) */
bool_t interactive = FALSE; /* set TRUE when main() is ready to roll */
char **files; /* list of input files */
int numfiles; /* number of input files */
int curfile; /* number of the current file */
#ifndef MACINTOSH
static void
usage()
{
fprintf(stderr, "usage: stevie [file ...]\n");
fprintf(stderr, " stevie -t tag\n");
fprintf(stderr, " stevie +[num] file\n");
fprintf(stderr, " stevie +/pat file\n");
exit(1);
}
#endif
main(argc,argv)
int argc;
char *argv[];
{
char *initstr, *getenv(); /* init string from the environment */
char *tag = NULL; /* tag from command line */
char *pat = NULL; /* pattern from command line */
int line = -1; /* line number from command line */
/*
* Process the command line arguments.
*/
if (argc > 1) {
switch (argv[1][0]) {
#ifndef MACINTOSH
case '-': /* -t tag */
if (argv[1][1] != 't')
usage();
if (argv[2] == NULL)
usage();
Filename = NULL;
tag = argv[2];
numfiles = 1;
break;
case '+': /* +n or +/pat */
if (argv[1][1] == '/') {
if (argv[2] == NULL)
usage();
Filename = strsave(argv[2]);
pat = &(argv[1][1]);
numfiles = 1;
} else if (isdigit(argv[1][1]) || argv[1][1] == NUL) {
if (argv[2] == NULL)
usage();
Filename = strsave(argv[2]);
numfiles = 1;
line = (isdigit(argv[1][1])) ?
atoi(&(argv[1][1])) : 0;
} else
usage();
break;
#endif
default: /* must be a file name */
Filename = strsave(argv[1]);
files = &(argv[1]);
numfiles = argc - 1;
break;
}
} else {
Filename = NULL;
numfiles = 1;
}
curfile = 0;
if (numfiles > 1)
fprintf(stderr, "%d files to edit\n", numfiles);
windinit();
/*
* Allocate LPTR structures for all the various position pointers
*/
if ((Filemem = (LPTR *) malloc(sizeof(LPTR))) == NULL ||
(Filetop = (LPTR *) malloc(sizeof(LPTR))) == NULL ||
(Fileend = (LPTR *) malloc(sizeof(LPTR))) == NULL ||
(Topchar = (LPTR *) malloc(sizeof(LPTR))) == NULL ||
(Botchar = (LPTR *) malloc(sizeof(LPTR))) == NULL ||
(Curschar = (LPTR *) malloc(sizeof(LPTR))) == NULL ||
(Insstart = (LPTR *) malloc(sizeof(LPTR))) == NULL ) {
fprintf(stderr, "Can't allocate data structures\n");
windexit(0);
}
screenalloc();
filealloc(); /* Initialize Filemem, Filetop, and Fileend */
screenclear();
if ((initstr = getenv("EXINIT")) != NULL) {
char *lp, buf[128];
if ((lp = getenv("LINES")) != NULL) {
sprintf(buf, "%s lines=%s", initstr, lp);
docmdln(buf);
} else
docmdln(initstr);
}
if (Filename != NULL) {
if (readfile(Filename, Filemem, FALSE))
filemess("[New File]");
} else if (tag == NULL)
msg("Empty Buffer");
setpcmark();
#ifndef MACINTOSH
if (tag) {
stuffin(":ta ");
stuffin(tag);
stuffin("\n");
} else if (pat) {
stuffin(pat);
stuffin("\n");
} else if (line >= 0) {
if (line > 0)
stuffnum(line);
stuffin("G");
}
#endif
interactive = TRUE;
edit();
windexit(0);
return 1; /* shouldn't be reached */
}
#define RBSIZE 1024
static char getcbuff[RBSIZE];
static char *getcnext = NULL;
void
stuffin(s)
char *s;
{
if (s == NULL) { /* clear the stuff buffer */
getcnext = NULL;
return;
}
if (getcnext == NULL) {
strcpy(getcbuff,s);
getcnext = getcbuff;
} else
strcat(getcbuff,s);
}
void
stuffnum(n)
int n;
{
char buf[32];
sprintf(buf, "%d", n);
stuffin(buf);
}
int
vgetc()
{
register int c;
/*
* inchar() may map special keys by using stuffin(). If it does
* so, it returns -1 so we know to loop here to get a real char.
*/
do {
if ( getcnext != NULL ) {
int nextc = *getcnext++;
if ( *getcnext == NUL ) {
*getcbuff = NUL;
getcnext = NULL;
}
return(nextc);
}
c = inchar();
} while (c == -1);
return c;
}
/*
* anyinput
*
* Return non-zero if input is pending.
*/
bool_t
anyinput()
{
return (getcnext != NULL);
}
/*
* do_mlines() - process mode lines for the current file
*
* Returns immediately if the "ml" parameter isn't set.
*/
#define NMLINES 5 /* no. of lines at start/end to check for modelines */
void
do_mlines()
{
void chk_mline();
int i;
register LPTR *p;
if (!P(P_ML))
return;
p = Filemem;
for (i=0; i < NMLINES ;i++) {
chk_mline(p->linep->s);
if ((p = nextline(p)) == NULL)
break;
}
if ((p = prevline(Fileend)) == NULL)
return;
for (i=0; i < NMLINES ;i++) {
chk_mline(p->linep->s);
if ((p = prevline(p)) == NULL)
break;
}
}
/*
* chk_mline() - check a single line for a mode string
*/
static void
chk_mline(s)
register char *s;
{
register char *cs; /* local copy of any modeline found */
register char *e;
for (; *s != NUL ;s++) {
if (strncmp(s, "vi:", 3) == 0 || strncmp(s, "ex:", 3) == 0) {
cs = strsave(s+3);
if ((e = strchr(cs, ':')) != NULL) {
*e = NUL;
stuffin(mkstr(CTRL('o')));
docmdln(cs);
}
free(cs);
}
}
}